Skip to main content

23.5. Application Events and Listeners

23.5. 应用事件和监听器

除了常见的Spring框架事件,比如ContextRefreshedEventSpringApplication也会发送其他的应用事件。

有些事件实际上是在ApplicationContext创建前触发的,所以你不能在那些事件(处理类)中通过@Bean注册监听器,只能通过SpringApplication.addListeners(…)SpringApplicationBuilder.listeners(…)方法注册。如果想让监听器自动注册,而不关心应用的创建方式,你可以在工程中添加一个META-INF/spring.factories文件,并使用org.springframework.context.ApplicationListener作为key指向那些监听器,如下:

org.springframework.context.ApplicationListener=com.example.project.MyListener

应用运行时,事件会以下面的次序发送:

  1. 在运行开始,但除了监听器注册和初始化以外的任何处理之前,会发送一个ApplicationStartingEvent
  2. 在Environment将被用于已知的上下文,但在上下文被创建前,会发送一个ApplicationEnvironmentPreparedEvent
  3. 在refresh开始前,但在bean定义已被加载后,会发送一个ApplicationPreparedEvent
  4. 在上下文更新后,但在任何应用和命令行运行器被调用前,会发送一个ApplicationStartedEvent
  5. 在任何应用和命令行运行器被调用后,会发送一个ApplicationReadyEvent,表示应用准备好接收请求了。
  6. 启动过程中如果出现异常,会发送一个ApplicationFailedEvent

通常不需要使用应用事件,但知道它们的存在是有用的(在某些场合可能会使用到),比如,在Spring Boot内部会使用事件处理各种任务。

应用事件通过Spring框架的事件发布机制发送。此机制的一部分确保了一个发布到子上下文里的监听器的事件,也会发布到任何祖先上下文里的监听器。因此,如果你的应用使用SpringApplication实例的层级,监听器可能会接收到应用事件的多个相同类型的实例。

为了让你的监听器区分一个事件是对应它的上下文还是子孙上下文,它应当请求它的应用上下文被注入,然后比较注入的上下文与事件的上下文。上下文可以通过实现ApplicationContextAware注入。或者,如果监听器是一个bean,可以使用@Autowired